package com.example.arithmeticleetcode.learnArithmetic2.sort;

/**
 * @program: arithmetic-leetcode
 * @description: 最好 最坏 平局时间复杂度为 O(nlogN) 稳定排序  空间复杂度为O(n)
 * @author: FangZhen
 * @create: 2020-11-06 15:17
 **/
public class MergeSort<E extends Comparable<E>> extends Sort<E>{

    private E[] leftArray;


    @Override
    protected void sort() {
        leftArray = (E[]) new Comparable[array.length >> 1];
        sort(0, array.length);
    }

    /**
     * 对[begin, end)范围的数据进行归并排序
     * @param begin
     * @param end
     */
    private void sort(int begin, int end) {
        if (end - begin < 2) return;

        int mid = (begin + end) >> 1;
        sort(begin, mid);
        sort(mid, end);
        merge(begin, mid, end);
    }

    /**
     * 将[begin, mid) 和 [mid, end)范围的序列合并成一个有序序列
     * @param begin
     * @param mid
     * @param end
     */
    private void merge(int begin, int mid, int end) {
        int li = 0, le = mid - begin;
        int ri = mid, re = end;
        int ai = begin;
        //备份左边数据
        for (int i = li; i < le; i++) {
            leftArray[i] = array[begin + i];
        }
        //如果左边还没有结束
        while (li < le) {
            if (ri < re && cmp(array[ri], leftArray[li]) < 0) {
                array[ai++] = array[ri++];
            } else {
                array[ai++] = leftArray[li++];
            }
        }
    }
}
